/*
 * Copyright 2010-2016 OrientDB LTD (http://orientdb.com)
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.orientechnologies.orient.test.database.auto;

import com.orientechnologies.common.concur.ONeedRetryException;
import com.orientechnologies.orient.core.db.ODatabaseSession;
import com.orientechnologies.orient.core.record.impl.ODocument;
import com.orientechnologies.orient.test.ConcurrentTestHelper;
import com.orientechnologies.orient.test.TestFactory;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicLong;
import org.testng.Assert;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Optional;
import org.testng.annotations.Parameters;
import org.testng.annotations.Test;

@Test
public class ConcurrentQueriesTest extends DocumentDBBaseTest {
  private static final int THREADS = 10;
  private static final int CYCLES = 50;
  private static final int MAX_RETRIES = 50;

  private final AtomicLong counter = new AtomicLong();
  private final AtomicLong totalRetries = new AtomicLong();

  @Parameters(value = "url")
  public ConcurrentQueriesTest(@Optional String url) {
    super(url);
  }

  class CommandExecutor implements Callable<Void> {

    String url;

    public CommandExecutor(String url) {
      this.url = url;
    }

    @Override
    public Void call() {
      for (int i = 0; i < CYCLES; i++) {
        try (ODatabaseSession db = openSession("admin", "admin")) {
          for (int retry = 0; retry < MAX_RETRIES; ++retry) {
            try {
              db.command("select from Concurrent").close();

              counter.incrementAndGet();
              totalRetries.addAndGet(retry);
              break;
            } catch (ONeedRetryException e) {
              try {
                Thread.sleep(retry * 10);
              } catch (InterruptedException e1) {
                throw new RuntimeException(e1);
              }
            }
          }
        }
      }
      return null;
    }
  }

  @BeforeClass
  public void init() {
    if (database.getMetadata().getSchema().existsClass("Concurrent"))
      database.getMetadata().getSchema().dropClass("Concurrent");

    database.getMetadata().getSchema().createClass("Concurrent");

    for (int i = 0; i < 1000; ++i) {
      database.save(database.<ODocument>newInstance("Concurrent").field("test", i));
    }
  }

  @Test
  public void concurrentCommands() throws Exception {
    //    System.out.println("Spanning " + THREADS + " threads...");

    ConcurrentTestHelper.test(
        THREADS,
        new TestFactory<Void>() {
          @Override
          public Callable<Void> createWorker() {
            return new CommandExecutor(url);
          }
        });

    //    System.out.println("Done! Total queries executed in parallel: " + counter.get() + "
    // average retries: "
    //        + ((float) totalRetries.get() / (float) counter.get()));

    Assert.assertEquals(counter.get(), CYCLES * THREADS);
  }
}
